Signature generation
Take
https://sapi.xt.com/v4/orderas an example.
The following appKey/secret are for demo only (Linux bash with echo, openssl, curl):
- appKey: 3976eb88-76d0-4f6e-a6b2-a57980770085
- secretKey: bc6630d0231fda5cd98794f52c4998659beda290
Required Headers
validate-algorithms: HmacSHA256
validate-appkey: 3976eb88-76d0-4f6e-a6b2-a57980770085
validate-recvwindow: 5000
validate-timestamp: 1641446237201
validate-signature: 2b5eb11e18796d12d88f13dc27dbbd02c2cc51ff7059765ed9821957d82bb4d9
Sample Request Body
{
  "type": "LIMIT",
  "timeInForce": "GTC",
  "side": "BUY",
  "symbol": "btc_usdt",
  "price": "39000",
  "quantity": "2"
}
1 Data Part Concatenation (Y)
- method: uppercase HTTP method, e.g. GET,POST,DELETE,PUT
- path: concrete RESTful path after filling variables, e.g. /sign/test/bb/aa
- query: sort all key=valueby key (lexicographical), join with&example:userName=dfdfdf&password=ggg
- body:
- JSON: use the raw JSON string (no conversion/sorting)
- x-www-form-urlencoded: sort all key=valueby key (lexicographical), join with&example:userName=dfdfdf&password=ggg
- form-data: not supported
 
- If multiple forms exist, concatenate in order: path→query→body.
Finally, splice by # with leading markers:
Y = #method#path#query#body
Notice:
- query present, body empty → Y = #method#path#query
- query empty, body present → Y = #method#path#body
- both present → Y = #method#path#query#body
Mini Examples
- 
Method: POST
- 
Path: /v4/order(recorded aspath)
- 
Query Example: symbol=btc_usdt(recorded asquery)
- 
Body Examples: - x-www-form-urlencoded- symbol=btc_usdt&side=BUY&type=LIMIT&timeInForce=GTC&quantity=1&price=0.1
- json- {"symbol":"btc_usdt","side":"BUY","type":"LIMIT","timeInForce":"GTC","quantity":2,"price":39000}
 
- 
Mixed (query + body/json) - query: symbol=btc_usdt&side=BUY&type=LIMIT
- body(json): {"symbol":"btc_usdt","side":BUY,"type":"LIMIT"}
 
- query: 
2 Header Part Concatenation (X)
Sort the following header keys in natural ascending alphabetical order, then join with &:
X = validate-algorithms=HmacSHA256&validate-appkey=3976eb88-76d0-4f6e-a6b2-a57980770085&validate-recvwindow=5000&validate-timestamp=1641446237201
3Generate Signature
Concatenate original = X + Y (no delimiter beyond the # already in Y), then compute:
signature = org.apache.commons.codec.digest.HmacUtils.hmacSha256Hex(secretKey, original);
Add the generated value to the request header:
validate-signature: <signature>
4 Complete Example
Sample original signature message:
validate-algorithms=HmacSHA256&validate-appkey=2063495b-85ec-41b3-a810-be84ceb78751&validate-recvwindow=60000&validate-timestamp=1666026215729#POST#/v4/order#{"symbol":"XT_USDT","side":"BUY","type":"LIMIT","timeInForce":"GTC","bizType":"SPOT","price":3,"quantity":2}
Sample request (cURL):
curl --location --request POST 'https://sapi.xt.com/v4/order' \
  --header 'accept: */*' \
  --header 'Content-Type: application/json' \
  --header 'validate-algorithms: HmacSHA256' \
  --header 'validate-appkey: 10c172ca-d791-4da5-91cd-e74d202dac96' \
  --header 'validate-recvwindow: 60000' \
  --header 'validate-timestamp: 1666026215729' \
  --header 'validate-signature: 4cb36e820f50d2e353e5e0a182dc4a955b1c26efcb4b513d81eec31dd36072ba' \
  --data-raw '{"symbol":"XT_USDT","side":"BUY","type":"LIMIT","timeInForce":"GTC","bizType":"SPOT","price":3,"quantity":2}'
Matters needing attention
- Ensure Content-Type, signature original message, and final request payload are consistent.
- validate-timestampshould be milliseconds of the send time; pair with a reasonable- validate-recvwindowto tolerate network jitter.
- When body is JSON, use the exact raw JSON string for signing (don’t reorder keys or prettify).